home
***
CD-ROM
|
disk
|
FTP
|
other
***
search
/
Amiga Format CD 24
/
Amiga Format AFCD24 (Feb 1998, Issue 108).iso
/
-in_the_mag-
/
emulation
/
uae
/
uae-0.4.3
/
newcpu.c
< prev
next >
Wrap
C/C++ Source or Header
|
1998-01-20
|
14KB
|
657 lines
/*
* UAE - The Un*x Amiga Emulator
*
* MC68000 emulation
*
* (c) 1995 Bernd Schmidt
*/
#include <stdio.h>
#include <assert.h>
#include <string.h>
#include <stdlib.h>
#include "config.h"
#include "amiga.h"
#include "options.h"
#include "events.h"
#include "memory.h"
#include "custom.h"
#include "newcpu.h"
#include "ersatz.h"
#include "readcpu.h"
cpuop_func *cpufunctbl[65536];
#undef COUNT_INSTRS
#ifdef COUNT_INSTRS
static unsigned long int instrcount[65536];
static UWORD opcodenums[65536];
int compfn(const void *el1, const void *el2)
{
return instrcount[*(const UWORD *)el1] < instrcount[*(const UWORD *)el2];
}
void dump_counts(void)
{
FILE *f = fopen("insncount", "w");
unsigned long int total = 0;
int i;
for(i=0; i < 65536; i++) {
opcodenums[i] = i;
total += instrcount[i];
}
qsort(opcodenums, 65536, sizeof(UWORD), compfn);
fprintf(f, "Total: %ld\n", total);
for(i=0; i < 65536; i++) {
unsigned long int cnt = instrcount[opcodenums[i]];
if (!cnt)
break;
fprintf(f, "%04x: %ld\n", opcodenums[i], cnt);
}
fclose(f);
}
#endif
bool broken_in;
#ifdef DUALCPU
bool allowmem;
bool customacc;
#endif
static cpuop_func *find_func_for (UWORD opcode)
{
int i = 0;
if (cpufunctbl[opcode] != op_illg)
return cpufunctbl[opcode];
for (;;i++) {
if (smallcputbl[i].opcode == opcode)
return smallcputbl[i].handler;
}
}
void init_m68k (void)
{
long int opcode;
#ifdef COUNT_INSTRS
int i;
for(i=0;i<65536;i++) {
instrcount[i] = 0;
}
#endif
read_table68k ();
do_merges ();
printf("Building CPU table...\n");
for (opcode = 0; opcode < 65536; opcode++)
cpufunctbl[opcode] = op_illg;
for (opcode = 0; opcode < 65536; opcode++) {
if (table68k[opcode].mnemo == i_ILLG)
continue;
if (table68k[opcode].handler == -1)
cpufunctbl[opcode] = find_func_for (opcode);
else
cpufunctbl[opcode] = find_func_for (table68k[opcode].handler);
}
}
struct regstruct regs;
static void ShowEA(int reg, amodes mode, wordsizes size)
{
UWORD dp;
BYTE disp8;
WORD disp16;
int r;
ULONG dispreg;
CPTR addr;
switch(mode){
case Dreg:
printf("D%d", reg);
break;
case Areg:
printf("A%d", reg);
break;
case Aind:
printf("(A%d)", reg);
break;
case Aipi:
printf("(A%d)+", reg);
break;
case Apdi:
printf("-(A%d)", reg);
break;
case Ad16:
disp16 = nextiword();
addr = regs.a[reg] + (WORD)disp16;
printf("(A%d,$%04lx) == $%08lx", reg, disp16, (long unsigned int)addr);
break;
case Ad8r:
dp = nextiword();
disp8 = dp & 0xFF;
r = (dp & 0x7000) >> 12;
dispreg = dp & 0x8000 ? regs.a[r] : regs.d[r];
if (!(dp & 0x800)) dispreg = (LONG)(WORD)(dispreg);
addr = regs.a[reg] + disp8 + dispreg;
printf("(A%d, %c%d.%c, $%02x) == $%08lx", reg,
dp & 0x8000 ? 'A' : 'D', (int)r, dp & 0x800 ? 'L' : 'W', disp8,
(long unsigned int)addr);
break;
case PC16:
addr = m68k_getpc();
disp16 = nextiword();
addr += (WORD)disp16;
printf("(PC,$%08lx) == $%08lx", disp16, (long unsigned int)addr);
break;
case PC8r:
addr = m68k_getpc();
dp = nextiword();
disp8 = dp & 0xFF;
r = (dp & 0x7000) >> 12;
dispreg = dp & 0x8000 ? regs.a[r] : regs.d[r];
if (!(dp & 0x800)) dispreg = (LONG)(WORD)(dispreg);
addr += disp8 + dispreg;
printf("(PC, %c%d.%c, $%02x) == $%08lx", dp & 0x8000 ? 'A' : 'D',
(int)r, dp & 0x800 ? 'L' : 'W', disp8, (long unsigned int)addr);
break;
case absw:
printf("$%08lx", (LONG)(WORD)nextiword());
break;
case absl:
printf("$%08lx", nextilong());
break;
case imm:
switch(size){
case sz_byte:
printf("#$%02x", nextiword() & 0xff); break;
case sz_word:
printf("#$%04x", nextiword()); break;
case sz_long:
printf("#$%08lx", nextilong()); break;
default:
abort();
}
break;
case imm0:
printf("#$%02lx", nextiword() & 0xff);
break;
case imm1:
printf("#$%04lx", nextiword());
break;
case imm2:
printf("#$%08lx", nextilong());
break;
case immi:
printf("#$%04lx", reg);
break;
default:
abort();
}
}
#if CPU_LEVEL > 1
ULONG get_disp_ea (ULONG base, UWORD dp)
{
int reg = (dp >> 12) & 7;
LONG regd;
if (dp & 0x8000)
regd = regs.a[reg];
else
regd = regs.d[reg];
if (!(dp & 0x800))
regd = (LONG)(WORD)regd;
if (dp & 0x100) {
LONG extraind = 0;
printf("020\n");
regd <<= (dp >> 9) & 3;
if (dp & 0x80)
base = 0;
if (dp & 0x40)
regd = 0;
if ((dp & 0x30) == 0x20)
base += (LONG)(WORD)nextiword();
if ((dp & 0x30) == 0x30)
base += nextilong();
if ((dp & 0x3) == 0x2)
extraind = (LONG)(WORD)nextiword();
if ((dp & 0x3) == 0x3)
extraind = nextilong();
if (!(dp & 4))
base += regd;
if (dp & 3)
base = get_long (base);
if (dp & 4)
base += regd;
return base + extraind;
/* Yikes, that's complicated */
} else {
return base + (BYTE)(dp) + regd;
}
}
#endif
static void SuperState(void)
{
if (!regs.s){
CPTR temp = regs.usp;
regs.s = 1;
regs.t = 0;
regs.usp = regs.a[7];
regs.a[7] = temp;
}
}
static void UserState(void)
{
if (regs.s){
CPTR temp = regs.usp;
regs.s = 0;
/* regs.t = 0; */
regs.usp = regs.a[7];
regs.a[7] = temp;
}
}
void MakeSR(void)
{
#if 0
assert((regs.n & 1) == regs.n);
assert((regs.s & 1) == regs.s);
assert((regs.x & 1) == regs.x);
assert((regs.c & 1) == regs.c);
assert((regs.v & 1) == regs.v);
assert((regs.z & 1) == regs.z);
#endif
regs.sr = ((regs.t << 15) | (regs.s << 13) | (regs.intmask << 8)
| (regs.x << 4) | (regs.n << 3) | (regs.z << 2) | (regs.v << 1)
| regs.c);
}
void MakeFromSR(void)
{
int olds = regs.s;
regs.t = (regs.sr >> 15) & 1;
regs.s = (regs.sr >> 13) & 1;
regs.intmask = (regs.sr >> 8) & 7;
regs.x = (regs.sr >> 4) & 1;
regs.n = (regs.sr >> 3) & 1;
regs.z = (regs.sr >> 2) & 1;
regs.v = (regs.sr >> 1) & 1;
regs.c = regs.sr & 1;
if (olds != regs.s) {
regs.s = olds;
if (regs.s) UserState(); else SuperState();
}
specialflags |= SPCFLAG_INT;
if (regs.t)
specialflags |= SPCFLAG_TRACE;
else
specialflags &= ~(SPCFLAG_TRACE | SPCFLAG_DOTRACE);
}
void Exception(int nr)
{
MakeSR();
SuperState();
if (CPU_LEVEL > 0) {
regs.a[7] -= 2;
put_word (regs.a[7], nr * 4);
}
regs.a[7] -= 4;
put_long (regs.a[7], m68k_getpc ());
regs.a[7] -= 2;
put_word (regs.a[7], regs.sr);
m68k_setpc(get_long(regs.vbr + 4*nr));
regs.t = 0;
specialflags &= ~(SPCFLAG_TRACE | SPCFLAG_DOTRACE);
}
static void Interrupt(int nr)
{
assert(nr < 8 && nr >= 0);
Exception(nr+24);
regs.intmask = nr;
specialflags |= SPCFLAG_INT;
}
static int caar, cacr;
void m68k_move2c (int regno, ULONG *regp)
{
if (CPU_LEVEL == 1 && (regno & 0x7FF) > 1)
op_illg (0x4E7B);
else
switch (regno) {
case 0: regs.sfc = *regp; break;
case 1: regs.dfc = *regp; break;
case 2: cacr = *regp & 0xFF; break;
case 0x800: regs.usp = *regp; break;
case 0x801: regs.vbr = *regp; break;
case 0x802: caar = *regp & 0xFF; break;
default:
op_illg (0x4E7B);
break;
}
}
void m68k_movec2 (int regno, ULONG *regp)
{
if (CPU_LEVEL == 1 && (regno & 0x7FF) > 1)
op_illg (0x4E7A);
else
switch (regno) {
case 0: *regp = regs.sfc; break;
case 1: *regp = regs.dfc; break;
case 2: *regp = cacr; break;
case 0x800: *regp = regs.usp; break;
case 0x801: *regp = regs.vbr; break;
case 0x802: *regp = caar; break;
default:
op_illg (0x4E7A);
break;
}
}
extern void m68k_divl (UWORD opcode, ULONG src, UWORD extra)
{
if (src == 0)
return;
#ifdef INT_64BIT
if (extra & 0x800) {
/* signed variant */
INT_64BIT a = regs.d[(extra >> 12) & 7];
INT_64BIT quot, rem;
if (extra & 0x400)
a |= (INT_64BIT)regs.d[extra & 7] << 32;
rem = a % src;
quot = a / src;
if ((extra & 0x400) && (extra & 7) != ((extra >> 12) & 7))
regs.d[extra & 7] = rem;
regs.d[(extra >> 12) & 7] = quot;
} else {
/* unsigned */
unsigned INT_64BIT a = regs.d[(extra >> 12) & 7];
unsigned INT_64BIT quot, rem;
if (extra & 0x400)
a |= (INT_64BIT)regs.d[extra & 7] << 32;
rem = a % src;
quot = a / src;
if ((extra & 0x400) && (extra & 7) != ((extra >> 12) & 7))
regs.d[extra & 7] = rem;
regs.d[(extra >> 12) & 7] = quot;
}
#endif
}
extern void m68k_mull (UWORD opcode, ULONG src, UWORD extra)
{
#ifdef INT_64BIT
if (extra & 0x800) {
/* signed variant */
INT_64BIT a = (LONG)regs.d[(extra >> 12) & 7];
a *= (LONG)src;
if ((extra & 0x400) && (extra & 7) != ((extra >> 12) & 7))
regs.d[extra & 7] = a >> 32;
regs.d[(extra >> 12) & 7] = (ULONG)a;
} else {
/* unsigned */
unsigned INT_64BIT a = (ULONG)regs.d[(extra >> 12) & 7];
unsigned INT_64BIT quot, rem;
a *= src;
if ((extra & 0x400) && (extra & 7) != ((extra >> 12) & 7))
regs.d[extra & 7] = a >> 32;
regs.d[(extra >> 12) & 7] = (ULONG)a;
}
#endif
}
static char* ccnames[] =
{ "T ","F ","HI","LS","CC","CS","NE","EQ",
"VC","VS","PL","MI","GE","LT","GT","LE" };
void MC68000_reset(void)
{
regs.a[7] = get_long(0x00f80000);
m68k_setpc(get_long(0x00f80004));
regs.s = 1;
regs.stopped = 0;
regs.t = 0;
specialflags = 0;
regs.intmask = 7;
regs.vbr = regs.sfc = regs.dfc = 0;
customreset();
}
void op_illg(UWORD opcode)
{
if (opcode == 0xF00D && ((m68k_getpc() & 0xF80000) == 0xF80000)) {
/* This is from the dummy Kickstart replacement */
ersatz_perform (nextiword ());
return;
}
regs.pc_p--;
if ((opcode & 0xF000) == 0xF000) {
if ((opcode & 0xE00) == 0x200)
Exception(0xB);
else
switch (opcode & 0x1FF) {
case 0x17:
regs.pc_p+=2;
break;
default:
regs.pc_p++;
}
return;
}
if ((opcode & 0xF000) == 0xA000) {
Exception(0xA);
return;
}
fprintf(stderr, "Illegal instruction: %04x\n", opcode);
Exception(4);
}
static int n_insns=0, n_spcinsns=0;
static __inline__ void do_hardware(void)
{
if (specialflags & SPCFLAG_BLIT) {
do_blitter();
#ifdef NO_FAST_BLITTER
do_blitter();
do_blitter();
do_blitter();
#endif
}
if (specialflags & SPCFLAG_DISK) {
do_disk(); /* This is not critical. Four calls make disk */
do_disk(); /* loading quite fast. */
do_disk();
do_disk();
}
}
void MC68000_run(void)
{
for(;;) {
UWORD opcode;
/* assert (!regs.stopped && !(specialflags & SPCFLAG_STOP)); */
opcode = nextiword();
#ifdef COUNT_INSTRS
instrcount[opcode]++;
#endif
(*cpufunctbl[opcode])(opcode);
#ifndef NO_EXCEPTION_3
if (buserr) {
Exception(3);
buserr = false;
}
#endif
/*n_insns++;*/
do_cycles();
if (specialflags) {
/*n_spcinsns++;*/
while (specialflags & SPCFLAG_STOP) {
do_cycles();
do_hardware();
if (specialflags & (SPCFLAG_INT | SPCFLAG_DOINT)){
int intr = intlev();
specialflags &= ~(SPCFLAG_INT | SPCFLAG_DOINT);
specialflags &= ~(SPCFLAG_INT | SPCFLAG_DOINT);
if (intr != -1 && intr > regs.intmask) {
Interrupt(intr);
regs.stopped = 0;
specialflags &= ~SPCFLAG_STOP;
}
}
}
if (specialflags & SPCFLAG_DOTRACE) {
Exception(9);
}
if (specialflags & SPCFLAG_TRACE) {
specialflags &= ~SPCFLAG_TRACE;
specialflags |= SPCFLAG_DOTRACE;
}
#ifdef WANT_SLOW_MULTIPLY
/* Kludge for Hardwired demo. The guys who wrote it should be
* mutilated. */
if (specialflags & SPCFLAG_EXTRA_CYCLES) {
do_cycles ();
do_cycles ();
do_cycles ();
do_cycles ();
specialflags &= ~SPCFLAG_EXTRA_CYCLES;
}
#endif
do_hardware();
if (specialflags & SPCFLAG_DOINT) {
int intr = intlev();
specialflags &= ~(SPCFLAG_INT | SPCFLAG_DOINT);
if (intr != -1 && intr > regs.intmask) {
Interrupt(intr);
regs.stopped = 0;
}
}
if (specialflags & SPCFLAG_INT) {
specialflags &= ~SPCFLAG_INT;
specialflags |= SPCFLAG_DOINT;
}
if (specialflags & SPCFLAG_BRK) {
specialflags &= ~SPCFLAG_BRK;
return;
}
}
}
}
void MC68000_step(void)
{
specialflags |= SPCFLAG_BRK;
MC68000_run();
}
void MC68000_skip(CPTR nextpc)
{
broken_in = false;
specialflags |= SPCFLAG_BRK;
do {
MC68000_step();
} while (nextpc != m68k_getpc() && !broken_in);
}
void MC68000_disasm(CPTR addr, CPTR *nextpc, int cnt)
{
CPTR pc = m68k_getpc();
m68k_setpc(addr);
for (;cnt--;){
char instrname[20],*ccpt;
int opwords;
UWORD opcode;
UWORD special = 0;
struct mnemolookup *lookup;
struct instr *dp;
printf("%08lx: ", m68k_getpc());
for(opwords = 0; opwords < 5; opwords++){
printf("%04x ", get_word(m68k_getpc() + opwords*2));
}
opcode = nextiword();
if (cpufunctbl[opcode] == op_illg) {
opcode = 0x4AFC;
}
dp = table68k + opcode;
for (lookup = lookuptab;lookup->mnemo != dp->mnemo; lookup++)
;
strcpy(instrname,lookup->name);
ccpt = strstr(instrname,"cc");
if (ccpt != 0) {
strncpy(ccpt,ccnames[dp->cc],2);
}
printf("%s", instrname);
switch(dp->size){
case sz_byte: printf(".B "); break;
case sz_word: printf(".W "); break;
case sz_long: printf(".L "); break;
default: break;
}
if (dp->suse) {
ShowEA(dp->sreg, dp->smode, dp->size);
}
if (dp->suse && dp->duse)
printf(",");
if (dp->duse) {
ShowEA(dp->dreg, dp->dmode, dp->size);
}
if (ccpt != 0) {
if (cctrue(dp->cc))
printf(" (TRUE)");
else
printf(" (FALSE)");
}
printf("\n");
}
*nextpc = m68k_getpc();
m68k_setpc(pc);
}
void MC68000_dumpstate(CPTR *nextpc)
{
int i;
for(i = 0; i < 8; i++){
printf("D%d: %08lx ", i, regs.d[i]);
if ((i & 3) == 3) printf("\n");
}
for(i=0;i<8;i++){
printf("A%d: %08lx ", i, regs.a[i]);
if ((i & 3) == 3) printf("\n");
}
printf ("T=%d S=%d X=%d N=%d Z=%d V=%d C=%d IMASK=%d\n", regs.t, regs.s,
regs.x, regs.n, regs.z, regs.v, regs.c, regs.intmask);
MC68000_disasm(m68k_getpc(), nextpc, 1);
printf("next PC: %08lx\n", *nextpc);
}